home *** CD-ROM | disk | FTP | other *** search
/ ftp.mactech.com 2010 / ftp.mactech.com.tar / ftp.mactech.com / thinkref / archive / THINKPascalUH2.1.sea / THINKPas Univ Hdr 2.1 / Interfaces / DisAsmLookup.p < prev    next >
Text File  |  1995-09-12  |  23KB  |  472 lines

  1. { Converted with MPW2TPas Tuesday, September 12, 1995 5:51:46 PM }
  2. {
  3.     File:        DisAsmLookup.p
  4.  
  5.     Copyright:    © 1983-1993 by Apple Computer, Inc.
  6.                 All rights reserved.
  7.  
  8.     Version:    System 7.1 for ETO #11
  9.     Created:    Tuesday, March 30, 1993 18:00
  10.  
  11. }
  12.  
  13.     UNIT DisAsmLookup;
  14.     INTERFACE
  15.  
  16.  
  17. {$IFC UNDEFINED UsingDisAsmLookup}
  18. {$SETC UsingDisAsmLookup := 1}
  19.  
  20.   USES
  21.    ConditionalMacros, Types;
  22. TYPE
  23.     LookupRegs    = (_A0_, _A1_, _A2_, _A3_, _A4_, _A5_, _A6_, _A7_,
  24.                                  _PC_, _ABS_, _TRAP_, _IMM_);
  25.  
  26. {----------------------------------------------------------------------}
  27.  
  28. PROCEDURE Disassembler(      DstAdjust:  LongInt;                  {addr correction}
  29.                                              VAR BytesUsed:  Integer;                  {bytes used up  }
  30.                                                      FirstByte:  UNIV Ptr;                 {starting byte  }
  31.                                              VAR Opcode:         UNIV Str255;          {mnemonic             }
  32.                                              VAR Operand:      UNIV Str255;        {operand        }
  33.                                              VAR Comment:      UNIV Str255;          {comment        }
  34.                                                      LookupProc: UNIV Ptr);              {search proc    }
  35.     {}
  36. {     Disassembler is a Pascal routine to be called to disassemble a sequence}
  37. {     of bytes.  All MC68xxx, MC68881, and MC68851 instructions are supported.}
  38. {     The sequence of bytes to be disassembled are pointed to by FirstByte.}
  39. {     BytesUsed bytes starting at FirstByte are consumed by the disassembly,}
  40. {     and the Opcode, Operand, and Comment strings returned as NULL TERMINATED}
  41. {     Pascal strings (for easier manipulation with C). The caller is then free}
  42. {     to format or use the output strings any way appropriate to the}
  43. {     application.}
  44. {     }
  45. {     Depending on the opcode and effective address(s) (EA's) to be}
  46. {     disassembled, the Opcode, Operand, and Comment strings contain the}
  47. {     following information:}
  48. {     }
  49. {     Case                     Opcode    Operand    Comment}
  50. {     =======================================================================}
  51. {     Non PC-relative EA's     op.sz     EA's             ; 'c…' (for immediates)}
  52. {     PC-relative EA's         op.sz     EA's       ; address}
  53. {     Toolbox traps            DC.W      $AXXX      ; TB XXXX}
  54. {     OS traps                 DC.W      $AXXX      ; OS XXXX}
  55. {     Invalid bytes            DC.W      $XXXX      ; ????}
  56. {     =======================================================================}
  57. {     }
  58. {     For valid disassembly of processor instructions the appropriate MC68XXX}
  59. {     opcode mnemonic is generated for the Opcode string along with a size}
  60. {     attribute when required.  The source and destination EA's are generated}
  61. {     as the Operand along with a possible comment. Comments start with a ';'.}
  62. {     Traps use a DC.W assembler directive as the Opcode with the trap word}
  63. {     as the Operand and a comment indicating whether the trap is a toolbox or}
  64. {     OS trap and what the trap number is.  As described later the caller can}
  65. {     generate symbolic substitutions into EA's and provide names for traps.}
  66. {    }
  67. {     Invalid instructions cause the string 'DC.W' to be returned in the}
  68. {     Opcode string. Operand is '$XXXX' (the invalid word) with a comment of}
  69. {     '; ????'.  BytesUsed is 2.  This is similar to the trap call case except}
  70. {     for the comment.}
  71. {    }
  72. {     Note, the Operand EA's is syntatically similar to but NOT COMPATIBLE}
  73. {     with the MPW assembler!    This is because the Disassembler generates}
  74. {     byte hex constants as "$XX" and word hex constants as "$XXXX".  Negative}
  75. {     values (e.g., $FF or $FFFF) produced by the Disassembler are treated as}
  76. {     long word values by the MPW assembler.  Thus it is assumed that}
  77. {     Disassembler output will NOT be used as MPW assembler input.  If that is}
  78. {     the goal, then the caller must convert strings of the form $XX or $XXXX}
  79. {     in the Operand string to their decimal equivalent.  The routine}
  80. {     ModifyOperand is provided in this unit to aid with the conversion}
  81. {     process.}
  82. {    }
  83. {     Since a PC-relative comment is an address, the only address that the}
  84. {     Disassembler knows about is the address of the code pointed to by}
  85. {     FirstByte.  Generally, that may be a buffer that has no relation to}
  86. {     "reality", i.e., the actual code loaded into the buffer.  Therefore,}
  87. {     to allow the address comment to be mapped back to some actual address}
  88. {     the caller may specify an adjustment factor, specified by DstAdjust,}
  89. {     that is ADDED to the value that normally would be placed in the}
  90. {     comment.}
  91. {    }
  92. {     Operand effective address strings are generated as a function of the }
  93. {     effective address mode and a special case is made for A-trap opcode}
  94. {     strings. In places where a possible symbolic reference could be}
  95. {     substituted for an address (or a portion of an address), the Disassembler}
  96. {     can call a user specified routine to do the substitution (using the}
  97. {     LookupProc parameter described later).  The following table summarizes}
  98. {     the generated effective addresses and where symbolic substitutions (S)}
  99. {     can be made:}
  100. {     }
  101. {     Mode    Generated Effective Address  Effective Address with Substitution}
  102. {     ========================================================================}
  103. {         0     Dn                           Dn}
  104. {         1     An                           An}
  105. {         2     (An)                         (An)}
  106. {         3     (An)+                        (An)+}
  107. {         4     -(An)                        -(An)}
  108. {         5     ∂(An)                        S(An) or just S (if An=A5, ∂≥0)}
  109. {        6n     ∂(An,Xn.Size*Scale)          S(An,Xn.Size*Scale)}
  110. {        6n     (BD,An,Xn.Size*Scale)        (S,An,Xn.Size*Scale)}
  111. {        6n     ([BD,An],Xm.Size*Scale,OD)   ([S,An],Xm.Size*Scale,OD)}
  112. {        6n     ([BD,An,Xn.Size*Scale],OD)   ([S,An,Xn.Size*Scale],OD)}
  113. {        70     ∂                            S}
  114. {        71     ∂                            S}
  115. {        72     *±∂                          S}
  116. {        73     *±∂(Xn.Size*Scale)           S(Xn.Size*Scale)}
  117. {        73     (*±∂,Xn.Size*Scale)          (S,Xn.Size*Scale)}
  118. {        73     ([*±∂],Xm.Size*Scale,OD)     ([S],Xm.Size*Scale,OD)}
  119. {        73     ([*±∂,Xn.Size*Scale],OD)     ([S,Xn.Size*Scale],OD)}
  120. {        74     #data                        S (#data made comment)}
  121. {     A-traps $AXXX                        S (as opcode, AXXX made comment)}
  122. {     ========================================================================}
  123. {}
  124. {     For A-traps, the substitution can be performed to substitute for the DC.W}
  125. {     opcode string.  If the substitution is made then the Disassembler will}
  126. {     generate ,Sys and/or ,Immed flags as operands for Toolbox traps and}
  127. {     ,AutoPop for OS traps when the bits in the trap word indicates these}
  128. {     settings.}
  129. {     }
  130. {                     |         Generated          |            Substituted}
  131. {                     | Opcode  Operand  Comment   | Opcode  Operand        Comment}
  132. {     ========================================================================}
  133. {     Toolbox | DC.W    $AXXX    ; TB XXXX | S       [,Sys][,Immed] ; AXXX}
  134. {     OS      | DC.W    $AXXX    ; OS XXXX | S       [,AutoPop]     ; AXXX}
  135. {     ========================================================================}
  136. {}
  137. {     All displacements (∂, BD, OD) are hexadecimal values shown as a byte}
  138. {     ($XX), word ($XXXX), or long ($XXXXXXXX) as appropriate.  The *Scale is}
  139. {     suppressed if 1. The Size is W or L.  Note that effective address}
  140. {     substitutions can only be made for "∂(An)", "BD,An", and "*±∂" cases.}
  141. {    }
  142. {     For all the effective address modes 5, 6n, 7n, and for A-traps, a}
  143. {     coroutine (a procedure) whose address is specified by the LookupProc}
  144. {     parameter is called by the Disassembler (if LookupProc is not NIL) to}
  145. {     do the substitution (or A-trap comment) with a string returned by the}
  146. {     proc.  It is assumed that the proc pointed to by LookupProc is a level 1}
  147. {     Pascal proc declared as follows:}
  148. {}
  149. {     PROCEDURE Lookup(        PC:      UNIV Ptr;     (*Addr of extension/trap word* )}
  150. {                                                BaseReg: LookupRegs;   (*Base register/lookup mode  * )}
  151. {                                                Opnd:    UNIV LongInt; (*Trap word, PC addr, disp.  * )}
  152. {                                        VAR S:       Str255);          (*Returned substitution      * )}
  153. {}
  154. {     or in C,}
  155. {     }
  156. {     pascal void LookUp(Ptr         PC,      /* Addr of extension/trap word */}
  157. {                                            LookupRegs  BaseReg, /* Base register/lookup mode   */}
  158. {                                            long        Opnd,    /* Trap word, PC addr, disp.   */}
  159. {                                            char              *S);     /* Returned substitution       */}
  160. {    }
  161. {     PC      = Pointer to instruction extension word or A-trap word in the}
  162. {                         buffer pointed to by the Disassembler's FirstByte parameter.}
  163. {}
  164. {     BaseReg = This determines the meaning of the Opnd value and supplies}
  165. {                         the base register for the "∂(An)", "BD,An", and "*±∂" cases.}
  166. {                         BaseReg may contain any one of the following values:}
  167. {                        }
  168. {                         _A0_    =  0 ==> A0}
  169. {                         _A1_    =  1 ==> A1}
  170. {                         _A2_    =  2 ==> A2}
  171. {                         _A3_    =  3 ==> A3}
  172. {                         _A4_    =  4 ==> A4}
  173. {                         _A5_    =  5 ==> A5}
  174. {                         _A6_    =  6 ==> A6}
  175. {                         _A7_    =  7 ==> A7}
  176. {                         _PC_    =  8 ==> PC-relative (special case)}
  177. {                         _ABS_   =  9 ==> Abs addr    (special case)}
  178. {                         _TRAP_  = 10 ==> Trap word   (special case)}
  179. {                         _IMM_     = 11 ==> Immediate   (special case)}
  180. {                         }
  181. {                         For absolute addressing (modes 70 and 71), BaseReg contains}
  182. {                         _ABS_.  For A-traps, BaseReg would contain _TRAP_.  For}
  183. {                         immediate data (mode 74), BaseReg would contain _IMM_.}
  184. {     }
  185. {     Opnd    = The contents of this LongInt is determined by the BaseReg}
  186. {                         parameter just described.}
  187. {}
  188. {                         For BaseReg = _IMM_ (immediate data):}
  189. {                                 Opnd contains the (extended) 32-bit immediate data specified}
  190. {                                 by the instruction.}
  191. {}
  192. {                         For BaseReg = _TRAP_ (A-traps):}
  193. {                                 Opnd is the entire trap word. The high order 16 bits of}
  194. {                                 Opnd are zero.}
  195. { }
  196. {                         For BaseReg = _ABS_  (absolute effective address):}
  197. {                                 Opnd contains the (extended) 32-bit address specifed by}
  198. {                                 the instruction's effective address.  Such addresses would}
  199. {                                 generally be used to reference low memory globals on a}
  200. {                                 Macintosh.}
  201. { }
  202. {                         For BaseReg = _PC_  (PC-relative effective address):}
  203. {                                 Opnd contains the 32-bit address represented by "*±∂"}
  204. {                                 adjusted by the Disassembler's DstAdjust parameter.}
  205. {                                 }
  206. {                         For BaseReg = _An_  (effective address with a base register):}
  207. {                                 Opnd contains the (sign-extended) 32-bit (base)}
  208. {                                 displacement from the instruction's effective address.}
  209. {                                 }
  210. {                                 In the Macintosh environment, a BaseReg specifying A5}
  211. {                                 implies either global data references or Jump Table}
  212. {                                 references. Positive Opnd values with an A5 BaseReg thus}
  213. {                                 mean Jump Table references, while a negative offset would}
  214. {                                 mean a global data reference.  Base registers of A6 or A7}
  215. {                                 would usually mean local data.}
  216. {    }
  217. {     S       = Pascal string returned from Lookup containing the effective}
  218. {                         address substitution string or a trap name for A-traps.  S is}
  219. {                         set to null PRIOR to calling Lookup.  If it is still null on}
  220. {                         return, the string is not used.  If not null, then for A-traps,}
  221. {                         the returned string is used as the opcode string. In all other}
  222. {                         cases the string is substituted as shown in the above table.}
  223. {    }
  224. {     Depending on the application, the caller has three choices on how to}
  225. {     use the Disassembler and an associated Lookup proc:}
  226. {    }
  227. {     (1). The caller can call just the Disassembler and provide his own Lookup}
  228. {                proc. In that case the calling conventions discussed above must be}
  229. {                followed.}
  230. {    }
  231. {     (2). The caller can provide NIL for the LookupProc parameter, in which}
  232. {                case, NO Lookup proc will be called.}
  233. {                }
  234. {     (3). The caller can call first InitLookup (described below, a proc}
  235. {                provided with this unit) and pass the address of this unit's}
  236. {                standard Lookup proc when Disassembler is called.    In this case all}
  237. {                the control logic to determine the kind of substitution to be done}
  238. {                is provided for the caller and all that need to be provided by the}
  239. {                user are routines to look up any or all of the following:}
  240. {                }
  241. {                • PC-relative references}
  242. {                • Jump Table references}
  243. {                • Absolute address references}
  244. {                • Trap names}
  245. {                • Immediate data names}
  246. {                • References with offsets from base registers                                            }
  247.                 
  248.  
  249. PROCEDURE InitLookup(PCRelProc, JTOffProc, TrapProc, AbsAddrProc, IdProc, ImmDataProc: UNIV Ptr);
  250.     {Prepare for use of this unit's Lookup proc.  When Disassembler is called
  251.      and the address of this unit's Lookup proc is specified, then for immediate
  252.      data, PC-relative, Jump Table references, A-traps, absolute addresses, and
  253.      offsets from a base register, the associated level 1 Pascal proc
  254.      specified here is called (if not NIL -- all six addresses are preset to
  255.      NIL).  The calls assume the following declarations for these procs (see
  256.      Lookup, below for further details):
  257.                                                      
  258.      PROCEDURE PCRelProc(Address: UNIV LongInt; 
  259.                                              VAR S:      UNIV Str255);
  260.  
  261.      PROCEDURE JTOffProc(A5JTOffset: UNIV Integer;
  262.                                              VAR S:         UNIV Str255);
  263.  
  264.      PROCEDURE TrapNameProc(TrapWord: UNIV Integer;
  265.                                                     VAR S:      UNIV Str255);
  266.                                                      
  267.      PROCEDURE AbsAddrProc(AbsAddr: UNIV LongInt;
  268.                                                  VAR S:      UNIV Str255);
  269.  
  270.      PROCEDURE IdProc(BaseReg: LookupRegs;
  271.                                         Offset:  UNIV LongInt;
  272.                                         VAR S:     UNIV Str255);
  273.                                                      
  274.      PROCEDURE ImmDataProc(ImmData: UNIV LongInt;
  275.                                                  VAR S:      UNIV Str255);
  276.  
  277.      Note: InitLookup contains initialized data which requires initializing
  278.                  at load time (this is of concern only to users with assembler
  279.                  main programs.}
  280.     
  281.  
  282.  PROCEDURE Lookup(        PC:      UNIV Ptr;     {Addr of extension/trap word}
  283.                                             BaseReg: LookupRegs;   {Base register/lookup mode  }
  284.                                             Opnd:    UNIV LongInt; {Trap word, PC addr, disp.  }
  285.                                     VAR S:       Str255);          {Returned substitution      }
  286.     {This is a standard Lookup proc available to the caller for calls to the
  287.      Disassembler.    If the caller elects to use this proc, then InitLookup
  288.      MUST be called prior to any calls to the Disassembler.  All the logic
  289.      to determine the type of lookup is done by this proc.  For PC-relative,
  290.      Jump Table references, A-traps, absolute addresses, and offsets from a
  291.      base register, the associated level 1 Pascal proc specified in the
  292.      InitLookup call (if not NIL) is called.
  293.     
  294.      This scheme simplifies the Lookup mechanism by allowing the caller
  295.      to deal with just the problems related to the application.}
  296.  
  297.  
  298. PROCEDURE LookupTrapName(TrapWord: UNIV Integer;
  299.                                                  VAR S:       UNIV Str255);
  300.     {This is a procedure provided to allow conversion of a trap instruction
  301.      (in TrapWord) to its corresponding trap name (in S).  It is provided
  302.      primarily for use with the Disassembler and its address may be passed to
  303.      InitLookup above for use by this unit's Lookup routine.  Alternatively,
  304.      there is nothing prohibiting the caller from using it directly for other
  305.      purposes or by some other Lookup proc.
  306.      
  307.      Note: The tables in this proc make the size of this proc about 9500
  308.                  bytes.  The trap names are fully spelled out in upper and lower
  309.                  case.}
  310.  
  311. PROCEDURE ModifyOperand(VAR Operand: UNIV Str255);
  312.     {Scan an operand string, i.e., the null terminated Pascal string returned
  313.      by the Disassembler (null MUST be present here) and modify negative hex
  314.      values to negated positive value.  For example, $FFFF(A5) would be
  315.      modified to -$0001(A5).  The operand to be processed is    passed as the
  316.      function's parameter which is edited "in place" and returned to the
  317.      caller.
  318.  
  319.      This routine is essentially a pattern matcher and attempts to only
  320.      modify 2, 4, and 8 digit hex strings in the operand that "might" be
  321.      offsets from a base register.  If the matching tests are passed, the
  322.      same number of original digits are output (because that indicates a
  323.      value's size -- byte, word, or long).
  324.  
  325.      For a hex string to be modified, the following tests must be passed:
  326.  
  327.      • There must have been exactly 2, 4, or 8 digits.
  328.  
  329.          Only hex strings $XX, $XXXX, and $XXXXXXXX are possible candidates
  330.          because that is the only way the Disassembler generates offsets.
  331.  
  332.      • Hex string must be delimited by a "(" or a ",".
  333.  
  334.          The "(" allows offsets for $XXXX(An,...) and $XX(An,Xn) addressing
  335.          modes.  The comma allows for the MC68020 addressing forms.
  336.  
  337.      • The "$X..." must NOT be preceded by a "±".
  338.  
  339.          This eliminates the possibility of modifying the offset of a
  340.          PC-relative addressing mode always generated in the form "*±$XXXX".
  341.      
  342.      • The "$X..." must NOT be preceded by a "#".
  343.      
  344.          This eliminates modifying immediate data.
  345.  
  346.      • Value must be negative.
  347.  
  348.          Negative values are the only values we modify.  A value $FFFF is
  349.          modified to -$0001.}
  350.  
  351. {FUNCTION validMacsBugSymbol(symStart, limit: UNIV Ptr;
  352.                                                         symbol: StringPtr): StringPtr; C;}
  353.     {Check that the bytes pointed to by symStart represents a valid MacsBug
  354.      symbol.  The symbol must be fully contained in the bytes starting at
  355.      symStart, up to, but not including, the byte pointed to by the limit
  356.      parameter.
  357.      
  358.      If a valid symbol is NOT found, then NIL is returned as the function's
  359.      result.  However, if a valid symbol is found, it is copied to symbol (if
  360.      it is not NIL) as a null terminated Pascal string, and return a pointer
  361.      to where we think the FOLLOWING module begins. In the "old style" cases
  362.      (see below) this will always be 8 or 16 bytes after the input symStart.
  363.      For new style Apple Pascal and C cases this will depend on the symbol
  364.      length, existence of a pad byte, and size of the constant (literal) area.
  365.      In all cases, trailing blanks are removed from the symbol.
  366.      
  367.      A valid MacsBug symbol consists of the characters '_', '%', spaces,
  368.      digits, and upper/lower case letters in a format determined by the first
  369.      two bytes of the symbol as follows:
  370.      
  371.         1st byte  | 2nd byte  |  Byte  |
  372.             Range   |  Range    | Length | Comments
  373.      =======================================================================
  374.         $20 - $7F | $20 - $7F |    8   | "Old style" MacsBug symbol format
  375.         $A0 - $7F | $20 - $7F |    8   | "Old style" MacsBug symbol format
  376.      -----------------------------------------------------------------------
  377.         $20 - $7F | $80 - $FF |   16   | "Old style" MacApp symbol ab==>b.a
  378.         $A0 - $7F | $80 - $FF |   16   | "Old style" MacApp symbol ab==>b.a
  379.      -----------------------------------------------------------------------
  380.         $80       | $01 - $FF |    n   | n = 2nd byte           (Apple symbol)
  381.         $81 - $9F | $00 - $FF |    m   | m = BAnd(1st byte,$7F) (Apple symbol)
  382.      =======================================================================
  383.      
  384.      The formats are determined by whether bit 7 is set in the first and
  385.      second bytes.  This bit will removed when we find it or'ed into the first
  386.      and/or second valid symbol characters.
  387.      
  388.      The first two formats in the above table are the basic "old style" (pre-
  389.      existing) MacsBug formats. The first byte may or may not have bit 7 set
  390.      the second byte is a valid symbol character.  The first byte (with bit 7
  391.      removed) and the next 7 bytes are assumed to comprise the symbol.
  392.      
  393.      The second pair of formats are also "old style" formats, but used for
  394.      MacApp symbols.  Bit 7 set in the second character indicates these
  395.      formats.  The symbol is assumed to be 16 bytes with the second 8 bytes
  396.      preceding the first 8 bytes in the generated symbol.  For example,
  397.      12345678abcdefgh represents the symbol abcdefgh.12345678.
  398.      
  399.      The last pair of formats are reserved by Apple and generated by the MPW
  400.      Pascal and C compilers.  In these cases the value of the first byte is
  401.      always between $80 and $9F, or with bit 7 removed, between $00 and $1F.
  402.      For $00, the second byte is the length of the symbol with that many bytes
  403.      following the second byte (thus a max length of 255). Values $01 to $1F
  404.      represent the length itself.  A pad byte may follow these variable length
  405.      cases if the symbol does not end on a word boundary.  Following the
  406.      symbol and the possible pad byte is a word containing the size of the
  407.      constants (literals) generated by the compiler.
  408.      
  409.      Note that if symStart actually does point to a valid MacsBug symbol,
  410.      then you may use showMacsBugSymbol to convert the MacsBug symbol bytes to
  411.      a string that could be used as a DC.B operand for disassembly purposes.
  412.      This string explicitly shows the MacsBug symbol encodings.}
  413.  
  414. {FUNCTION endOfModule(address, limit: UNIV Ptr; symbol: StringPtr;
  415.                                          VAR nextModule: UNIV Ptr): StringPtr; C;}
  416.     {Check to see if the specified memory address, contains a RTS, JMP (A0) or
  417.      RTD #n instruction immediately followed by a valid MacsBug symbol.  These
  418.      sequences are the only ones which can determine an end of module when
  419.      MacsBug symbols are present.  During the check, the instruction and its
  420.      following MacsBug symbol must be fully contained in the bytes starting at
  421.      the specified address parameter, up to, but not including, the byte
  422.      pointed to by the limit parameter.
  423.     
  424.      If the end of module is NOT found, then NIL is returned as the
  425.      function's result.  However, if a end of module is found, the MacsBug
  426.      symbol is returned in symbol (if it is not NIL) as a null terminated
  427.      Pascal string (with trailing blanks removed), and the functions returns
  428.      the pointer to the start of the MacsBug symbol (i.e., address+2 for RTS
  429.      or JMP (A0) and address+4 for RTD #n).  This address may then be used as
  430.      in input parameter to showMacsBugSymbol to convert the MacsBug symbol to
  431.      a Disassembler operand string.
  432.      
  433.      Also returned in nextModule is where we think the FOLLOWING module
  434.      begins. In the "old style" cases (see validMacsBugSymbol) this will
  435.      always be 8 or 16 bytes after the input address.  For new style the
  436.      Apple Pascal and C cases this will depend on the symbol length, existence
  437.      of a pad byte, and size of the constant (literal) area.  See 
  438.      validMacsBugSymbol for a description of valid MacsBug symbol formats.}
  439.     
  440. {FUNCTION showMacsBugSymbol(symStart, limit: UNIV Ptr; operand: StringPtr;
  441.                                                      VAR bytesUsed: Integer): StringPtr; C;}
  442.     {Format a MacsBug symbol as a operand of a DC.B directive.  The first one
  443.      or two bytes of the symbol are generated as $80+'c' if they have there
  444.      high high bits set.  All other characters are shown as characters in a
  445.      string constant.  The pad byte, if present, is one is also shown as $00.
  446.      
  447.      When called, showMacsBugSymbol assumes that symStart is pointing at a
  448.      valid MacsBug symbol as validated by the validMacsBugSymbol or
  449.      endOfModule routines. As with validMacsBugSymbol, the symbol must be
  450.      fully contained in the bytes starting at symStart up to, but not
  451.      including, the byte pointed to by the limit parameter.
  452.      
  453.      The string is returned in the 'operand' parameter as a null terminated
  454.      Pascal string.  The function also returns a pointer to this string as its
  455.      return value (NIL is returned only if the byte pointed to by the limit
  456.      parameter is reached prior to processing the entire symbol -- which
  457.      should not happen if properly validated).  The number of bytes used for
  458.      the symbol is returned in bytesUsed.  Due to the way MacsBug symbols are
  459.      encoded, bytesUsed may not necessarily be the same as the length of the
  460.      operand string.
  461.      
  462.      A valid MacsBug symbol consists of the characters '_', '%', spaces,
  463.      digits, and upper/lower case letters in a format determined by the first
  464.      two bytes of the symbol as described in the validMacsBugSymbol routine.}
  465.  
  466. {$ENDC}    { UsingDisAsmLookup }
  467.  
  468.     IMPLEMENTATION
  469.     END.
  470.  
  471.  
  472.